<html lang="zh-CN"><head><meta charset="UTF-8"><style>.nodata  main {width:1000px;margin: auto;}</style></head><body class="nodata " style=""><div class="main_father clearfix d-flex justify-content-center " style="height:100%;"> <div class="container clearfix " id="mainBox"><main><div class="blog-content-box">
<div class="article-header-box">
<div class="article-header">
<div class="article-title-box">
<h1 class="title-article" id="articleContentId">(A卷,200分)- 快速开租建站（Java & JS & Python）</h1>
</div>
</div>
</div>
<div id="blogHuaweiyunAdvert"></div>

        <div id="article_content" class="article_content clearfix">
        <link rel="stylesheet" href="https://csdnimg.cn/release/blogv2/dist/mdeditor/css/editerView/kdoc_html_views-1a98987dfd.css">
        <link rel="stylesheet" href="https://csdnimg.cn/release/blogv2/dist/mdeditor/css/editerView/ck_htmledit_views-044f2cf1dc.css">
                <div id="content_views" class="htmledit_views">
                    <h4 id="main-toc">题目描述</h4> 
<p>当前IT部门支撑了子公司颗粒化业务&#xff0c;该部门需要实现为子公司快速开租建站的能力&#xff0c;建站是指在一个全新的环境部署一套IT服务。</p> 
<ul><li>每个站点开站会由一系列部署任务项构成&#xff0c;每个任务项部署完成时间都是固定和相等的&#xff0c;设为1。</li><li>部署任务项之间可能存在依赖&#xff0c;假如任务2依赖任务1&#xff0c;那么等任务1部署完&#xff0c;任务2才能部署。</li><li>任务有多个依赖任务则需要等所有依赖任务都部署完该任务才能部署。</li><li>没有依赖的任务可以并行部署&#xff0c;优秀的员工们会做到完全并行无等待的部署。</li></ul> 
<p>给定一个站点部署任务项和它们之间的依赖关系&#xff0c;请给出一个站点的最短开站时间。</p> 
<p></p> 
<h4 id="%E8%BE%93%E5%85%A5%E6%8F%8F%E8%BF%B0">输入描述</h4> 
<p>第一行是任务数taskNum,第二行是任务的依赖关系数relationsNum</p> 
<p>接下来 relationsNum 行&#xff0c;每行包含两个id&#xff0c;描述一个依赖关系&#xff0c;格式为&#xff1a;IDi IDj&#xff0c;表示部署任务i部署完成了&#xff0c;部署任务j才能部署&#xff0c;IDi 和 IDj 值的范围为&#xff1a;[0, taskNum)</p> 
<p>注&#xff1a;输入保证部署任务之间的依赖不会存在环。</p> 
<p></p> 
<h4 id="%E8%BE%93%E5%87%BA%E6%8F%8F%E8%BF%B0">输出描述</h4> 
<p>一个整数&#xff0c;表示一个站点的最短开站时间。</p> 
<p></p> 
<h4>备注</h4> 
<ul><li>1 &#xff1c; taskNum ≤ 100</li><li>1 ≤ relationsNum ≤ 5000</li></ul> 
<p></p> 
<h4 id="%E7%94%A8%E4%BE%8B">用例</h4> 
<table border="1" cellpadding="1" cellspacing="1" style="width:500px;"><tbody><tr><td style="width:86px;">输入</td><td style="width:412px;">5<br /> 5<br /> 0 4<br /> 1 2<br /> 1 3<br /> 2 3<br /> 2 4</td></tr><tr><td style="width:86px;">输出</td><td style="width:412px;">3</td></tr><tr><td style="width:86px;">说明</td><td style="width:412px;"> <p>有5个部署任务项&#xff0c;5个依赖关系&#xff0c;如下图所示。</p> <p><img alt="" height="261" src="https://img-blog.csdnimg.cn/0c4070f92f4b4922a629b76c68e96bbb.png" width="338" /></p> <p>我们可以先同时部署任务项0和任务项1&#xff0c;然后部署任务项2&#xff0c;最后同时部署任务型3和任务型4.最短开战时间为3。</p> </td></tr></tbody></table> 
<table border="1" cellpadding="1" cellspacing="1" style="width:500px;"><tbody><tr><td style="width:86px;">输入</td><td style="width:412px;">5<br /> 3<br /> 0 3<br /> 0 4<br /> 1 3</td></tr><tr><td style="width:86px;">输出</td><td style="width:412px;">2</td></tr><tr><td style="width:86px;">说明</td><td style="width:412px;"> <p>有5个部署任务项&#xff0c;3个依赖关系&#xff0c;如下图所示。</p> <p><img alt="" height="196" src="https://img-blog.csdnimg.cn/af8bc75aa1ed4d179851dc68f37bf2e1.png" width="309" /></p> <p>我们可以先同时部署任务型0&#xff0c;任务型1&#xff0c;任务项2。然后再同时部署任务型4.最短开站时间为2。</p> </td></tr></tbody></table> 
<h4 id="%E9%A2%98%E7%9B%AE%E8%A7%A3%E6%9E%90">题目解析</h4> 
<p>本题是经典的拓扑排序问题。因为题目种各任务之间有较为明显的前后依赖关系。</p> 
<p>对于拓扑排序&#xff0c;我们可以看下面博客的解析&#xff1a;</p> 
<p><a href="https://blog.csdn.net/qfc_128220/article/details/127804547?spm&#61;1001.2014.3001.5502" title="LeetCode - 207 课程表_伏城之外的博客-CSDN博客">LeetCode - 207 课程表_伏城之外的博客-CSDN博客</a></p> 
<p><a href="https://fcqian.blog.csdn.net/article/details/127710834" rel="nofollow" title="华为OD机试 - 跳格子游戏_伏城之外的博客-CSDN博客_跳格子 华为机试">华为OD机试 - 跳格子游戏_伏城之外的博客-CSDN博客_跳格子 华为机试</a></p> 
<p></p> 
<p>本题要求的并非是让我们判断拓扑结构中是否有环结构&#xff0c;因为题目已经说明了</p> 
<blockquote> 
 <p>输入保证部署任务之间的依赖不会存在环</p> 
</blockquote> 
<p>本题是要我们求解最短开战时间&#xff0c;这里最短开战时间其实就是剥离入度0点的次数&#xff0c;如下图所示&#xff1a;</p> 
<p></p> 
<p>用例1中各顶点的入度值已用红色值标出</p> 
<p><img alt="" height="269" src="https://img-blog.csdnimg.cn/dc2c733e0ccf421d960b1c39b57bce10.png" width="304" /></p> 
<p></p> 
<p>我们可以不断地剥离入度为0地顶点&#xff0c;因为题目说&#xff1a;</p> 
<blockquote> 
 <p>没有依赖的任务可以并行部署&#xff0c;优秀的员工们会做到完全并行无等待的部署。</p> 
</blockquote> 
<p><img alt="" height="269" src="https://img-blog.csdnimg.cn/dc2c733e0ccf421d960b1c39b57bce10.png" width="304" /></p> 
<p><img alt="" height="206" src="https://img-blog.csdnimg.cn/967fcec0bfac4eeeb921b4b018d57ef5.png" width="267" /></p> 
<p> <img alt="" height="196" src="https://img-blog.csdnimg.cn/9481e9ade55f4e5caff1985dbbd330e8.png" width="298" /></p> 
<p>可以只需要剥离三次&#xff0c;而这也是最短建站时间。 </p> 
<p></p> 
<h4 id="%E7%AE%97%E6%B3%95%E6%BA%90%E7%A0%81">JavaScript算法源码</h4> 
<pre><code class="language-javascript">/* JavaScript Node ACM模式 控制台输入获取 */
const readline &#61; require(&#34;readline&#34;);

const rl &#61; readline.createInterface({
  input: process.stdin,
  output: process.stdout,
});

const lines &#61; [];
let taskNum, relationsNum;
rl.on(&#34;line&#34;, (line) &#61;&gt; {
  lines.push(line);

  if (lines.length &#61;&#61;&#61; 2) {
    taskNum &#61; lines[0] - 0;
    relationsNum &#61; lines[1] - 0;
  }

  if (relationsNum &amp;&amp; lines.length &#61;&#61;&#61; relationsNum &#43; 2) {
    const relations &#61; lines.slice(2).map((line) &#61;&gt; line.split(&#34; &#34;).map(Number));
    console.log(getResult(relations, taskNum));
    lines.length &#61; 0;
  }
});

function getResult(relations, taskNum) {
  const next &#61; {};
  const inDegree &#61; new Array(taskNum).fill(0);

  for (let relation of relations) {
    const [a, b] &#61; relation; // a → b
    next[a] ? next[a].push(b) : (next[a] &#61; [b]); // a的下一个顶点有b
    inDegree[b]&#43;&#43;; // b顶点的入度&#43;&#43;
  }

  const queue &#61; [];

  let t &#61; 1;
  for (let i &#61; 0; i &lt; taskNum; i&#43;&#43;) {
    if (inDegree[i] &#61;&#61;&#61; 0) {
      queue.push([i, t]); // i含义是入度为0的顶点&#xff0c;t含义是该顶点所处建站时间
    }
  }

  while (queue.length) {
    const [task, time] &#61; queue.shift(); // 注意这里为了维持t&#xff0c;一定要使用队列先进先出&#xff0c;出队代表删除某顶点

    next[task]?.forEach((nxt) &#61;&gt; {
      // 该顶点被删除&#xff0c;则其后继点的入度值--&#xff0c;若--后入度为0&#xff0c;则将成为新的出队点
      if (--inDegree[nxt] &#61;&#61;&#61; 0) {
        t &#61; time &#43; 1; // 此时建站时间&#43;1
        queue.push([nxt, t]);
      }
    });
  }

  return t;
}
</code></pre> 
<p></p> 
<h4>Java算法源码</h4> 
<pre><code class="language-java">import java.util.ArrayList;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.Scanner;

public class Main {
  public static void main(String[] args) {
    Scanner sc &#61; new Scanner(System.in);

    int taskNum &#61; sc.nextInt();
    int relationsNum &#61; sc.nextInt();

    int[][] relations &#61; new int[relationsNum][2];
    for (int i &#61; 0; i &lt; relationsNum; i&#43;&#43;) {
      relations[i][0] &#61; sc.nextInt();
      relations[i][1] &#61; sc.nextInt();
    }

    System.out.println(getResult(relations, taskNum));
  }

  public static int getResult(int[][] relations, int taskNum) {
    HashMap&lt;Integer, ArrayList&lt;Integer&gt;&gt; next &#61; new HashMap&lt;&gt;();
    int[] inDegree &#61; new int[taskNum];

    for (int[] relation : relations) {
      int a &#61; relation[0];
      int b &#61; relation[1];

      next.putIfAbsent(a, new ArrayList&lt;&gt;());
      next.get(a).add(b); // a的下一个顶点有b
      inDegree[b]&#43;&#43;; // b顶点的入度&#43;&#43;
    }

    LinkedList&lt;Integer[]&gt; queue &#61; new LinkedList&lt;&gt;();
    int t &#61; 1;

    for (int i &#61; 0; i &lt; taskNum; i&#43;&#43;) {
      if (inDegree[i] &#61;&#61; 0) {
        queue.add(new Integer[] {i, t}); // i含义是入度为0的顶点&#xff0c;t含义是该顶点所处建站时间
      }
    }

    while (queue.size() &gt; 0) {
      Integer[] tmp &#61; queue.removeFirst(); // 注意这里为了维持t&#xff0c;一定要使用队列先进先出&#xff0c;出队代表删除某顶点
      int task &#61; tmp[0];
      int time &#61; tmp[1];

      if (next.containsKey(task) &amp;&amp; next.get(task).size() &gt; 0) {
        for (Integer nxt : next.get(task)) {
          // 该顶点被删除&#xff0c;则其后继点的入度值--&#xff0c;若--后入度为0&#xff0c;则将成为新的出队点
          if (--inDegree[nxt] &#61;&#61; 0) {
            t &#61; time &#43; 1; // 此时建站时间&#43;1
            queue.add(new Integer[] {nxt, t});
          }
        }
      }
    }

    return t;
  }
}
</code></pre> 
<p></p> 
<h4>Python算法源码</h4> 
<pre><code class="language-python">taskNum &#61; int(input())
relationsNum &#61; int(input())

relations &#61; []
for i in range(relationsNum):
    relations.append(list(map(int, input().split())))

next &#61; {}
inDegree &#61; [0 for i in range(taskNum)]

for relation in relations:
    a, b &#61; relation
    # a的下一个顶点有b
    if next.get(a) is None:
        next[a] &#61; [b]
    else:
        next[a].append(b)
    # b顶点的入度&#43;&#43;
    inDegree[b] &#43;&#61; 1

queue &#61; []

t &#61; 1
for i in range(taskNum):
    if inDegree[i] &#61;&#61; 0:
        queue.append((i, t))  # i含义是入度为0的顶点&#xff0c;t含义是该顶点所处建站时间

while len(queue) &gt; 0:
    task, time &#61; queue.pop(0)  # 注意这里为了维持t&#xff0c;一定要使用队列先进先出&#xff0c;出队代表删除某顶点

    if next.get(task) is not None and len(next[task]) &gt; 0:
        for nxt in next[task]:
            # 该顶点被删除&#xff0c;则其后继点的入度值--&#xff0c;若--后入度为0&#xff0c;则将成为新的出队点
            inDegree[nxt] -&#61; 1
            if inDegree[nxt] &#61;&#61; 0:
                t &#61; time &#43; 1  # 此时建站时间&#43;1
                queue.append((nxt, t))

print(t)
</code></pre>
                </div>
        </div>
        <div id="treeSkill"></div>
        <div id="blogExtensionBox" style="width:400px;margin:auto;margin-top:12px" class="blog-extension-box"></div>
    <script>
  $(function() {
    setTimeout(function () {
      var mathcodeList = document.querySelectorAll('.htmledit_views img.mathcode');
      if (mathcodeList.length > 0) {
        for (let i = 0; i < mathcodeList.length; i++) {
          if (mathcodeList[i].naturalWidth === 0 || mathcodeList[i].naturalHeight === 0) {
            var alt = mathcodeList[i].alt;
            alt = '\\(' + alt + '\\)';
            var curSpan = $('<span class="img-codecogs"></span>');
            curSpan.text(alt);
            $(mathcodeList[i]).before(curSpan);
            $(mathcodeList[i]).remove();
          }
        }
        MathJax.Hub.Queue(["Typeset",MathJax.Hub]);
      }
    }, 1000)
  });
</script>
</div></html>